//package com.yu.init.bimq;
//
//import com.rabbitmq.client.Channel;
//import com.yu.init.common.ErrorCode;
//import com.yu.init.constant.ChartConstant;
//import com.yu.init.exception.BusinessException;
//import com.yu.init.exception.ThrowUtils;
//import com.yu.init.manager.AiManager;
//import com.yu.init.manager.BaiDuAiManager;
//import com.yu.init.model.entity.Chart;
//import com.yu.init.service.ChartService;
//
//import lombok.SneakyThrows;
//import lombok.extern.slf4j.Slf4j;
//import org.apache.commons.lang3.StringUtils;
//import org.redisson.api.RedissonClient;
//import org.springframework.amqp.rabbit.annotation.RabbitListener;
//import org.springframework.amqp.support.AmqpHeaders;
//import org.springframework.messaging.handler.annotation.Header;
//import org.springframework.stereotype.Component;
//
//import javax.annotation.Resource;
//import java.util.HashMap;
//import java.util.Map;
//
//@Component
//@Slf4j
//public class BiMessageConsumer {
//
//    @Resource
//    private ChartService chartService;
//    @Resource
//    private WebSocketServer webSocketServer;
//    @Resource
//    private AiManager aiManager;
//    @Resource
//    private BaiDuAiManager baiDuAiManager;
//    @Resource
//    private RedissonClient redissonClient;
//
//    /**
//     * @param message     接收到的消息内容。
//     * @param channel     与 RabbitMQ 通信的通道对象。
//     * @param deliveryTag 消息的交付标签，用来唯一标识消息的标签,用于手动确认消息。初始值为1
//     */
//    // 指定程序监听的消息队列和确认机制
//    @SneakyThrows //用于简化在方法中抛出异常的代码
//    @RabbitListener(queues = {BiMqConstant.BI_QUEUE_NAME}, ackMode = "MANUAL")//ackMode = "MANUAL"：指定消息确认模式为手动确认，即需要在方法中手动调用确认或拒绝消息的方法,确保消息在被正确处理后才被确认
//    public void receiveMessage(String message, Channel channel, @Header(AmqpHeaders.DELIVERY_TAG) long deliveryTag) {
//
//        log.info("receiveMessage message = {}", message);
//        if (StringUtils.isBlank(message)) {
//            // 如果失败，消息拒绝
//            //第一个布尔值 false：只想拒绝当前无法处理的消息，而不是其他消息。如果为 true，表示拒绝所有比当前消息具有更小或相同交付标签的消息。
//            //第二个布尔值 false：消息不会重新排队，而是被直接丢弃或移动到死信队列（当前配置是直接丢弃），如果为 true，消息会被重新放回队列，可以重新被消费者获取。
//            channel.basicNack(deliveryTag, false, false);
//            throw new BusinessException(ErrorCode.SYSTEM_ERROR, "消息为空");
//        }
//        String[] split = message.split("_");
//        String strChartId = split[0];
//        String strUserId = split[1];
//
//        long chartId = Long.parseLong(strChartId);
//        long userId = Long.parseLong(strUserId);
//
//        Chart chart = chartService.getById(chartId);
//        if (chart == null) {
//            channel.basicNack(deliveryTag, false, false);
//            throw new BusinessException(ErrorCode.NOT_FOUND_ERROR, "图表为空");
//        }
//        // 先修改图表任务状态为 “执行中”。等执行成功后，修改为 “已完成”、保存执行结果；执行失败后，状态修改为 “失败”，记录任务失败信息。
//        Chart updateChart = new Chart();
//        updateChart.setId(chart.getId());
//        updateChart.setStatus("running");
//        boolean b = chartService.updateById(updateChart);
//        if (!b) {
//            channel.basicNack(deliveryTag, false, false);
//            handleChartUpdateError(chart.getId(), "更新图表执行中状态失败");
//            return;
//        }
//        // 调用 AI
//        //String result = aiManager.dochat(ChartConstant.YU_MODEL_ID, buildUserInput(chart), userId);
//        String result = baiDuAiManager.chatSingle(ChartConstant.BAIDU_MODEL_ID, buildBaiduUserInput(chart), userId);
//
//        String[] splits = result.split("【【【【【");
//        if (splits.length < 3) {
//            //此处由于生成时间比较长，所以只重试一次。由于不抛异常，且使用MANUAL确认模式，因此不采用配置的方式设置重试。
//            log.info("准备重新分析...");
//            if (deliveryTag > 1) {
//                channel.basicNack(deliveryTag, false, false);//放入死信队列
//            }
//            channel.basicNack(deliveryTag, false, true);
//            handleChartUpdateError(chart.getId(), "AI 生成错误");
//            return;
//        }
//        String genChart = splits[1].trim();
//        String genResult = splits[2].trim();
//        Chart updateChartResult = new Chart();
//        updateChartResult.setId(chart.getId());
//        updateChartResult.setGenChart(genChart);
//        updateChartResult.setGenResult(genResult);
//        updateChartResult.setStatus(ChartConstant.SUCCEED_STATUS);
//        updateChartResult.setExecMessage("AI生成成功");
//        boolean updateResult = chartService.updateById(updateChartResult);
//        if (!updateResult) {
//            channel.basicNack(deliveryTag, false, false);
//            handleChartUpdateError(chart.getId(), "更新图表成功状态失败");
//        }
//        webSocketServer.sendToAllClient("图表生成好啦，快去看看吧！");
//        // 消息确认，第二个参数表示当前消息已成功被消费者处理，不再需要重新发送给其他消费者，因此可以从队列中删除。
//        channel.basicAck(deliveryTag, false);
//    }
//
//    /**
//     * 构建用户输入(鱼聪明)
//     *
//     * @param chart
//     * @return
//     */
//    private String buildUserInput(Chart chart) {
//        String goal = chart.getGoal();
//        String chartType = chart.getChartType();
//        String csvData = chart.getChartData();
//
//        // 构造用户输入
//        StringBuilder userInput = new StringBuilder();
//
//        //构建提问内容
//        if (StringUtils.isNotBlank(chartType)) {
//            userInput.append("分析目标：").append(goal + "请使用：").append(chartType).append("\n");
//        } else {
//            userInput.append("分析目标：").append(goal).append("\n");
//        }
//        userInput.append("原始数据：").append("\n");
//        userInput.append(csvData).append("\n");
//
//        return userInput.toString();
//    }
//
//    /**
//     * 构建用户输入(文心一言)
//     *
//     * @param chart
//     * @return
//     */
//    private Map<String, String> buildBaiduUserInput(Chart chart) {
//
//        String goal = chart.getGoal();
//        String chartType = chart.getChartType();
//        String csvData = chart.getChartData();
//        ThrowUtils.throwIf(StringUtils.isBlank(goal), ErrorCode.PARAMS_ERROR, "未指定分析目标");
//        //要分析的csv数据
//        Map map = new HashMap();
//        if (StringUtils.isNotBlank(chartType)) {
//            map.put("goal", goal + ",请使用：" + chartType);
//        } else {
//            map.put("goal", goal);
//        }
//        map.put("data", csvData);
//        return map;
//    }
//
//
//    private void handleChartUpdateError(long chartId, String execMessage) {
//        Chart updateChartResult = new Chart();
//        updateChartResult.setId(chartId);
//        updateChartResult.setStatus(ChartConstant.FAILED_STATUS);
//        updateChartResult.setExecMessage(execMessage);
//        boolean updateResult = chartService.updateById(updateChartResult);
//        webSocketServer.sendToAllClient("糟糕，好像出了点问题~");
//        if (!updateResult) {
//            log.error("更新图表失败状态失败" + chartId + "," + execMessage);
//        }
//    }
//}
